let copyVue = null;

class VueRouter {
  // Vue.use传入 VueRouter 的时候会调用 VueRouter 下的 install 静态方法
  // 调用install的时候会传两个参数 1、Vue的构造函数 2、可选的选项对象

  static install(Vue) {
    // console.log(this); // 这里的this是VueRouter的构造函数 所以不能调用init 只有实例才能
    // 1. 判断当前插件是否已经安装
    // 在install静态方法中设置一个installed属性记录是否已经执行过install
    if (VueRouter.install.installed) { return; }
    VueRouter.install.installed = true;
    // 2. 把Vue的构造函数记录到全局变量
    copyVue = Vue;
    // 3. 把创建Vue实例时传入的router对象注入到所有的Vue实例上
    // 用混入的方式在vue事件的生命周期函数中添加 在beforeCreate生命周期函数中
    // this肯定是指向 Vue 的实例 此时可以找到 $options
    copyVue.mixin({
      beforeCreate() {
        // 如果是组件就不执行 如果是vue实例就执行 组件的 options 里没有 router 这个字段
        // 因为往vue原型上挂载$router只需要执行一次就行
        if (this.$options.router) {
          console.log(this); // 这个this就是Vue的实例
          console.log(copyVue.prototype); // 这个是Vue的构造函数 copyVue.prototype 是原型
          // 把这个实例的router上挂载到Vue原型上让每一个Vue的实例都有这个router
          copyVue.prototype.$router = this.$options.router;
          copyVue.prototype.name = 'lanpang 666';
          this.$options.router.init(); // 接line9 所以要在实例化之后再调用init
        }
      },
    });
  }

  constructor(options) {
    this.options = options;
    this.routeMap = {};
    // data 是一个响应式的对象
    // Vue 中提供的 observable 可以创建一个响应式对象
    // 这个响应式对象可以直接用在渲染函数和计算属性里面
    this.data = copyVue.observable({
      // 当前路由地址 变化时router-view 中 const el = self.routeMap[self.data.current];
      // 所以router-view render的组件依赖于这个响应式数据 所以也会发生变化
      current: '/',
    });
  }

  init() {
    this.createRouteMap();
    this.initComponents(copyVue);
    this.initEvent();
  }

  // createRouteMap 把构造函数中 传过来的选项中的 rules 转换成键值对的形式存储到 routeMap对象中去
  createRouteMap() {
    this.options.routes.forEach((route) => {
      this.routeMap[route.path] = route.component;
    });
  }

  // 这个函数是实现 router-link router-view 两个组件
  initComponents(Vue) {
    const self = this;

    Vue.component('router-link', {
      props: {
        to: String,
      },
      // template: `<a :href="to"><slot></slot></a>`,
      render(h) {
        return h('a', {
          attrs: {
            href: this.to,
          },
          on: {
            click: this.clickHandler,
          },
        }, [this.$slots.default]);
      },
      methods: {
        clickHandler(e) {
          // console.log(this); // 这里是VueComponent
          window.history.pushState({}, '', this.to);
          this.$router.data.current = this.to;
          e.preventDefault();
        },
      },
    });

    Vue.component('router-view', {
      render(h) {
        const el = self.routeMap[self.data.current];
        return h(el);
      },
    });
  }

  // 注册 popstate 事件 处理浏览器上点击前进后退
  initEvent() {
    window.addEventListener('popstate', () => {
      console.log('initEvent');
      this.data.current = window.location.pathname;
    });
  }
}

export default VueRouter;
